查看原文
其他

Spring 为啥默认把bean设计成单例的?这篇讲的明明白白的

小小木 Java之间 2020-10-16

点击上方Java之间”,选择“置顶或者星标”

你关注的就是我关心的!

来源:juejin.im/post/5cab7ebf518825177637b2f9

上一篇:java中double类型精度丢失问题及解决方法

熟悉Spring开发的朋友都知道Spring提供了5种scope分别是singleton、prototype、request、session、global session。

如下图是官方文档上的截图,感兴趣的朋友可以进去看看这五种分别有什么不同。今天要介绍的是这五种中的前两种,也是Spring最初提供的bean scope singleton 和 prototype。

Spring官方文档介绍如下图:

更多内容可以看官方文档介绍,非常详细:

https://docs.spring.io/spring/docs/3.0.0.M3/reference/html/ch04s04.html


单例bean与原型bean的区别

如果一个bean被声明为单例的时候,在处理多次请求的时候在Spring容器里只实例化出一个bean,后续的请求都公用这个对象,这个对象会保存在一个map里面。当有请求来的时候会先从缓存(map)里查看有没有,有的话直接使用这个对象,没有的话才实例化一个新的对象,所以这是个单例的。但是对于原型(prototype)bean来说当每次请求来的时候直接实例化新的bean,没有缓存以及从缓存查的过程。

1.画图分析

2.源码分析

生成bean时先判断单例的还是原型的

如果是单例的则先尝试从缓存里获取,没有在新创建

结论:

  1. 单例的bean只有第一次创建新的bean 后面都会复用该bean,所以不会频繁创建对象。

  2. 原型的bean每次都会新创建


单例bean的优势

由于不会每次都新创建新对象所以有一下几个性能上的优势:

1.减少了新生成实例的消耗

新生成实例消耗包括两方面,第一,spring会通过反射或者cglib来生成bean实例这都是耗性能的操作,其次给对象分配内存也会涉及复杂算法。

2.减少jvm垃圾回收

由于不会给每个请求都新生成bean实例,所以自然回收的对象少了。

3.可以快速获取到bean

因为单例的获取bean操作除了第一次生成之外其余的都是从缓存里获取的所以很快。

有关bean实例化相关可以看着篇文章:

https://juejin.im/post/5ca42bfa6fb9a05e17799e07


单例bean的劣势

单例的bean一个很大的劣势就是他不能做到线程安全!!!,由于所有请求都共享一个bean实例,所以这个bean要是有状态的一个bean的话可能在并发场景下出现问题,而原型的bean则不会有这样问题(但也有例外,比如他被单例bean依赖),因为给每个请求都新创建实例。关于这方面我正在准备写一篇文章,在整理当中,感兴趣的朋友可以关注我,我后续写一篇详细的文章。


总结

Spring 为啥把bean默认设计成单例?

答案:为了提高性能!!!从几个方面,

  1. 少创建实例

  2. 垃圾回收

  3. 缓存快速获取

单例有啥劣势?

如果是有状态的话在并发环境下线程不安全

最近热文阅读:
1、java中double类型精度丢失问题及解决方法
2、14 款牛逼的 IDEA 插件,让你开发速度飞起来!
3、JVM 发生 OOM 的 8 种原因、及解决办法
4、我把SpringBoot项目从18.18M瘦身到0.18M,部署起来真省事!
5、面试必问的一致性Hash在负载均衡中的应用
6、Java开发必须掌握的 20+ 种 Spring 常用注解
7、来了来了!Chrome 高级玩法,秒变摸鱼神器(提前尝鲜还未发布的功能)
8、可怕!3.83 亿开房记录遭泄露,这家酒店竟然又被爆泄露520万用户数据了
9、盘点 10 个代码重构的小技巧
10、为什么代码规范要求SQL语句不要过多的join?
10、建议收藏 | Redis 使用 10 个小技巧
关注公众号,你想要的Java都在这里

    您可能也对以下帖子感兴趣

    文章有问题?点此查看未经处理的缓存